package com.haojin.java;

import org.junit.Test;

/**
 * @author .29.
 * @create 2022-04-01 17:57
 */
//String的使用
public class StringTest {
    /*
    结论：
    1.常量与常量的拼接结果在常量池中。且常量池中不会存在相同内容的常量
    2.只要其中有一个是变量，结果就在堆中。
    3.如果结果的结果调用intern()，返回值就在常量池中。



     */
@Test
public void test3(){
    String s1 = "javaEE";
    String s2 = "hadoop";

    String s3 = "javaEEhadoop";//字面量，在字符串常量池中
    String s4 = "javaEE"+"hadoop";//字面量的连接，也在字符串常量池中
    //除字面量外还有 变量 参与，相当于对象指向了 堆空间 中的地址值
    String s5 = s1+"hadoop";
    String s6 = "javaEE"+ s2;
    String s7 = s1+ s2;

    System.out.println(s3==s4);//true
    System.out.println(s3==s5);//false
    System.out.println(s3==s6);//false
    System.out.println(s3==s7);//false
    System.out.println(s5==s6);//false
    System.out.println(s5==s7);//false

    String s8 = s5.intern();//返回值得到的s8使用的 常量池 中已经存在的“javaEEhadoop”
    System.out.println(s3 == s8);//true

}









    /*
    String的实例化的方式：
    方式一.通过字面量定义的方式
    方式二.通过new + 构造器的方式

    面试题：String s = new String("abc");方式创建对象，在内存中创建了几个对象？
            两个：一个是堆空间中new的结构，另一个char[]对应常量池中的数据："abc"

     */
@Test
public void test2(){
    //此时s1和s2的数据javaEE声明在方法区中的字符串常量池中
    String s1 = "javaEE";
    String s2 = "javaEE";
    //通过new + 构造器的方式：此时的s3和s4保存的数据的地址值，是数据在堆空间中开辟空间以后对应的地址值
    String s3 = new String("javaEE");
    String s4 = new String("javaEE");

    System.out.println(s1 == s2);
    System.out.println(s4 == s3);
    System.out.println("*************");

    Person p1 = new Person("Tom",12);
    Person p2 = new Person("Tom",12);

    System.out.println(p1.name.equals(p2.name));//true
    System.out.println(p1.name == p2.name);//true

    p1.name = "Jerry";
    System.out.println(p2.name);//Tom

}






    /*
    String:字符串，使用一对""引起来表示
    1.String是声明为final的,不可被继承
    2.String 实现了Serializable接口：表示字符串时支持序列化的。
             实现了Comparable接口：表示String可以比较大小
    3.String内部定义了final char[] value用于存储字符串数据
    4.String：代表不可变的字符串序列。简称：不可变性
        体现：1.当对字符串重新赋值的时候，需要重写指定内存区域赋值，不能使用原有的value进行赋值。
             2.当对现有的字符串进行连接操作时，也需要重新指定内存区域赋值，不能使用原有的value赋值。
             3.当调用String的replace()修改指定字符或字符串时，也必须重新指定内存区域赋值，不能使用原有的value进行赋值。
    5.通过字面量的方式（区别于new）给一个字符串赋值，此时的字符串声明在字符串常量池当中
    6.字符串常量池当中，不会存放两个相同的字符串
     */
    @Test
    public void test(){

        String s1 = "abc";//字面量的定义方式
        String s2 = "abc";

        System.out.println(s1 == s2);//比较s1和s2的地址值
        //表明s1和s2指向同一个地址（方法区中的字符串常量池不会存放两个相同的字符串）

        s1 = "hello";
        System.out.println(s1 == s2);

        System.out.println(s1);//hello
        System.out.println(s2);//abc
        System.out.println("*************");

        String s3 = "abc";
        s3 +="def";
        System.out.println(s3);//abcdef
        System.out.println(s2);//abc

        System.out.println("*************");
        String s4 = "abc";
        String s5 = s4.replace('a', 'm');
        System.out.println(s4);//abc（s4没有变）
        System.out.println(s5);//mbc


    }
}
